aboutsummaryrefslogtreecommitdiff
path: root/web/pw-server/src/routes/matches/[match_id].svelte
blob: 1d18375e13a4ad59ce96b3748399b00fb8ab9b3e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
<script lang="ts" context="module">
  import { ApiClient } from "$lib/api_client";
  export async function load({ params, fetch }) {
    try {
      const matchId = params["match_id"];
      const apiClient = new ApiClient(fetch);
      const matchData = await apiClient.get(`/api/matches/${matchId}`);
      return {
        props: {
          matchData,
        },
      };
    } catch (error) {
      return {
        status: error.status,
        error: error,
      };
    }
  }
</script>

<script lang="ts">
  import { onMount } from "svelte";
  import Visualizer from "$lib/components/Visualizer.svelte";
  import PlayerLog from "$lib/components/PlayerLog.svelte";
  import Select from "svelte-select";
  import { PLAYER_COLORS } from "$lib/constants";
  import { currentUser } from "$lib/stores/current_user";

  export let matchLog: string | undefined;
  export let matchData: object;

  onMount(async () => {
    const apiClient = new ApiClient();
    matchLog = await apiClient.getText(`/api/matches/${matchData["id"]}/log`);
  });

  $: playersWithVisibleLog = matchData["players"]
    .map((player: any, index: number) => ({
      color: PLAYER_COLORS[index],
      value: index,
      playerId: index + 1, // stoopid player number + 1
      displayName: player["bot_name"] || "player",
      matchPlayer: player,
    }))
    .filter((item) => canSeePlayerLog($currentUser, item.matchPlayer));

  // TODO: refactor match logs so that users can no longer get match logs for other players.
  function canSeePlayerLog(user: object | null, matchPlayer: object): boolean {
    if (!matchPlayer["owner_id"]) {
      return true;
    }

    return matchPlayer["owner_id"] === user?.["user_id"];
  }

  // using the same value here causes svelte to freeze
  let dropdownSelectedPlayer: any;
  let selectedPlayer: any;
  $: if (playersWithVisibleLog.length == 1) {
    selectedPlayer = playersWithVisibleLog[0];
  } else {
    selectedPlayer = dropdownSelectedPlayer;
  }
</script>

<div class="container">
  <Visualizer {matchLog} {matchData} />
  <div class="output-pane">
    <div class="player-select">
      {#if playersWithVisibleLog.length == 1}
        <h3 class="player-log-header">
          player log for
          <span style:color={selectedPlayer["color"]}>{selectedPlayer.displayName}</span>
        </h3>
      {:else}
        <Select
          items={playersWithVisibleLog}
          label="displayName"
          clearable={false}
          searchable={false}
          bind:value={dropdownSelectedPlayer}
          placeholder="Select player to see logs"
        >
          <div slot="item" let:item>
            <span style:color={item.color}>{item.displayName}</span>
          </div>
        </Select>
      {/if}
    </div>
    <div class="player-log">
      <PlayerLog {matchLog} playerId={selectedPlayer?.["playerId"]} />
    </div>
  </div>
</div>

<style lang="scss">
  @use "src/styles/variables";
  .container {
    display: flex;
    // these are needed for making the visualizer fill the screen.
    min-height: 0;
    flex-grow: 1;
    overflow: hidden;
  }

  .player-select {
    padding: 0 20px;
  }

  .player-log-header {
    color: #eee;
  }

  .player-log {
    padding: 15px;
    overflow-y: scroll;
  }

  .output-pane {
    width: 600px;
    // overflow: hidden;
    display: flex;
    flex-direction: column;
    background-color: variables.$bg-color;
  }
</style>